import time
from datetime import datetime, timedelta

from flask import current_app, jsonify
from flask import g
from flask import redirect
from flask import render_template
from flask import request
from flask import session

from info import constants
from info.models import User, News, Category
from info.utils.commons import user_login_data
from info.utils.image_storage import image_storage
from info.utils.response_code import RET
from . import admin_blue

#功能描述: 获取分类列表
# 请求路径: /admin/news_category
# 请求方式: GET
# 请求参数: GET,无
# 返回值:GET,渲染news_type.html页面, data数据
@admin_blue.route('/news_category')
def news_category():
    #1,获取分类数据
    try:
        categories = Category.query.all()
        categories.pop(0)
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DATAERR,errmsg="获取分类失败")

    #2,携带分类数据渲染页面
    return render_template("admin/news_type.html",categories = categories)





# 功能描述: 新闻编辑详情
# 请求路径: /admin/news_edit_detail
# 请求方式: GET, POST
# 请求参数: GET, news_id, POST(news_id,title,digest,content,index_image,category_id)
# 返回值:GET,渲染news_edit_detail.html页面,data字典数据, POST(errno,errmsg)
@admin_blue.route('/news_edit_detail', methods=['GET', 'POST'])
def news_edit_detail():
    # １,如果是GET请求,获取参数
    if request.method == "GET":
        news_id = request.args.get("news_id")

        # 1.2,校验参数
        if not news_id:
            return jsonify(errno=RET.PARAMERR,errmsg="参数不全")

        # 1.3,取出新闻对象，判断新闻对象是否存在
        try:
            news = News.query.get(news_id)
            categories = Category.query.all()
            categories.pop(0)
        except Exception as e:
            current_app.logger.error(e)
            return jsonify(errno=RET.DBERR,errmsg="获取新闻失败")

        # 将分类列表转换为字典列表
        category_list = []
        for category in categories:
            category_list.append(category.to_dict())

        if not news:
            return jsonify(errno=RET.NODATA,errmsg="新闻不存在")

        # 1.4,携带参数，渲染页面
        return render_template("admin/news_edit_detail.html",news = news.to_dict(),category_list = category_list)

    # - 2.获取参数
    news_id = request.form.get("news_id")
    title = request.form.get("title")
    digest = request.form.get("digest")
    content = request.form.get("content")
    index_image = request.files.get("index_image")
    category_id = request.form.get("category_id")

    # - 3.校验参数,为空校验
    if not all([news_id,title,digest,content,index_image,category_id]):
        return jsonify(errno=RET.PARAMERR,errmsg="参数不全")

    # - 4.根据新闻编号,获取新闻对象
    try:
        news = News.query.get(news_id)
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR,errmsg="获取新闻失败")

    if not news:
        return jsonify(errno=RET.NODATA,errmsg="新闻不存在")

    # - 5.上传图片
    try:
       image_name = image_storage(index_image.read())
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.THIRDERR,errmsg="七牛云异常")

    # - 6.判断图片是否上传成功
    if not image_name:
        return jsonify(errno=RET.NODATA,errmsg="图片上传失败")

    # - 7.设置编辑信息,到新闻对象
    news.title = title
    news.digest = digest
    news.index_image_url = constants.QINIU_DOMIN_PREFIX + image_name
    news.content = content
    news.category_id = category_id

    # - 8.返回响应
    return jsonify(errno=RET.OK,errmsg="修改成功")

# 功能描述: 新闻编辑列表
# 请求路径: /admin/news_edit
# 请求方式: GET
# 请求参数: GET, p, keywords
# 返回值:GET,渲染news_edit.html页面,data字典数据
@admin_blue.route('/news_edit')
def news_edit():
    """
    １,获取参数
    2,校验参数
    3,参数类型转换
    4,分页查询
    5,设置分页属性，总页数，当前页，当前页对象
    6,将对象列表转换为字典列表
    7,携带参数渲染页面
    :return:
    """
    # １,获取参数
    page = request.args.get("p","1")
    keywords = request.args.get("keywords")

    # 2,参数类型转换
    try:
        page = int(page)
    except Exception as e:
        current_app.logger.error(e)
        page = 1

    # 3,分页查询
    try:

        filters = []
        if keywords:
            filters.append(News.title.contains(keywords))

        paginate = News.query.filter(*filters).order_by(News.create_time.desc()).paginate(page, 10, False)
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR,errmsg="获取用户失败")

    # 4,设置分页属性，总页数，当前页，当前页对象
    totalPage = paginate.pages
    currentPage = paginate.page
    items = paginate.items

    # 5,将对象列表转换为字典列表
    news_list = []
    for news in items:
        news_list.append(news.to_review_dict())

    # 6,携带参数渲染页面
    data = {
        "totalPage":totalPage,
        "currentPage":currentPage,
        "news_list":news_list
    }

    return render_template("admin/news_edit.html",data = data)

# 功能描述: 新闻审核详情
# 请求路径: /admin/news_review_detail
# 请求方式: GET,POST
# 请求参数: GET, news_id, POST,news_id, action
# 返回值:GET,渲染news_review_detail.html页面,data字典数据

@admin_blue.route('/news_review_detail', methods=['GET', 'POST'])
def news_review_detail():
    """
    1,如果是GET请求,获取参数
    1.2,校验参数
    1.3,取出新闻对象，判断新闻对象是否存在
    1.4,携带参数，渲染页面
    2,post请求，获取参数
    3,校验参数
    4,判断操作类型
    5,根据新闻编号，取出新闻对象
    6,判断新闻对象是否存在
    7,根据操作类型，改变新闻状态
    8,返回响应
    :return:
    """
    # １,如果是GET请求,获取参数
    if request.method == "GET":
        news_id = request.args.get("news_id")

        # 1.2,校验参数
        if not news_id:
            return jsonify(errno=RET.PARAMERR,errmsg="参数不全")

        # 1.3,取出新闻对象，判断新闻对象是否存在
        try:
            news = News.query.get(news_id)
        except Exception as e:
            current_app.logger.error(e)
            return jsonify(errno=RET.DBERR,errmsg="获取新闻失败")

        if not news:
            return jsonify(errno=RET.NODATA,errmsg="新闻不存在")

        # 1.4,携带参数，渲染页面
        return render_template("admin/news_review_detail.html",news = news.to_dict())

    # 2,post请求，获取参数
    news_id = request.json.get("news_id")
    action = request.json.get("action")

    # 3,校验参数
    if not all([news_id, action]):
        return jsonify(errno=RET.PARAMERR,errmsg="参数不全")

    # 4,判断操作类型
    if not action in ("accept","reject"):
        return jsonify(errno=RET.DBERR,errmsg="操作类型错误")

    # 5,根据新闻编号，取出新闻对像
    try:
        news = News.query.get(news_id)
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR,errmsg="获取新闻失败")

    # 6,判断新闻对象是否存在
    if not news:
        return jsonify(errno=RET.NODATA,errmsg="新闻不存在")

    # 7,根据操作类型，改变新闻状态
    if action == "accept":
        news.status = 0
    else:
        reason = request.json.get("reason","")
        news.reason = reason
        news.status = -1

    # 8,返回响应
    return jsonify(errno=RET.OK,errmsg="审核成功")

# 功能描述: 新闻审核列表
# 请求路径: /admin/news_review
# 请求方式: GET
# 请求参数: GET, p
# 返回值:渲染news_review页面,data字典数据

@admin_blue.route('/news_review')
def news_review():
    """
    １,获取参数
    2,校验参数
    3,参数类型转换
    4,分页查询
    5,设置分页属性，总页数，当前页，当前页对象
    6,将对象列表转换为字典列表
    7,携带参数渲染页面
    :return:
    """
    # １,获取参数
    page = request.args.get("p","1")
    keywords = request.args.get("keywords")

    # 2,参数类型转换
    try:
        page = int(page)
    except Exception as e:
        current_app.logger.error(e)
        page = 1

    # 3,分页查询
    try:

        filters = [News.status != 0]
        if keywords:
            filters.append(News.title.contains(keywords))

        paginate = News.query.filter(*filters).order_by(News.create_time.desc()).paginate(page, 10, False)
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR,errmsg="获取用户失败")

    # 4,设置分页属性，总页数，当前页，当前页对象
    totalPage = paginate.pages
    currentPage = paginate.page
    items = paginate.items

    # 5,将对象列表转换为字典列表
    news_list = []
    for news in items:
        news_list.append(news.to_review_dict())

    # 6,携带参数渲染页面
    data = {
        "totalPage":totalPage,
        "currentPage":currentPage,
        "news_list":news_list
    }

    return render_template("admin/news_review.html",data = data)

#功能描述: 用户列表统计
# 请求路径: /admin/user_list
# 请求方式: GET
# 请求参数: p
# 返回值:渲染user_list.html页面,data字典数据

@admin_blue.route('/user_list')
def user_list():
    """
    １,获取参数
    2,校验参数
    3,参数类型转换
    4,分页查询
    5,设置分页属性，总页数，当前页，当前页对象
    6,将对象列表转换为字典列表
    7,携带参数渲染页面
    :return:
    """
    # １,获取参数
    page = request.args.get("p","1")

    # 2,参数类型转换
    try:
        page = int(page)
    except Exception as e:
        current_app.logger.error(e)
        page = 1

    # 3,分页查询
    try:
        paginate = User.query.filter(User.is_admin == False).order_by(User.create_time.desc()).paginate(page, 10, False)
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR,errmsg="获取用户失败")

    # 4,设置分页属性，总页数，当前页，当前页对象
    totalPage = paginate.pages
    currentPage = paginate.page
    items = paginate.items

    # 5,将对象列表转换为字典列表
    user_list = []
    for user in items:
        user_list.append(user.to_admin_dict())

    # 6,携带参数渲染页面
    data = {
        "totalPage":totalPage,
        "currentPage":currentPage,
        "user_list":user_list
    }

    return render_template("admin/user_list.html",data = data)

# 功能：用户统计
# 请求路径: /admin/user_count
# 请求方式: GET
# 请求参数: 无
# 返回值:渲染页面user_count.html,字典数据
@admin_blue.route('/user_count')
def user_count():
    """
    1,统计用户总人数
    2,统计月活人数
    3,统计月活人数
    4,时间段内活跃人数
    5,携带参数,渲染页面
    :return:
    """
    # 1,统计用户总人数
    try:
        total_count = User.query.filter(User.is_admin == False).count()
    except Exception as e:
        current_app.logger.error(e)

    # 2,统计月活人数
    cal = time.localtime()
    try:
        # 2.1本月１日０点
        month_startTime_str = "%d-%d-01"%(cal.tm_year,cal.tm_mon)
        # strptime() 参数１：表示时间字符串　　参数２：表示时间格式
        month_startTime_date = datetime.strptime(month_startTime_str,"%Y-%m-%d")

        # 2.2　此刻时间
        month_endTime_date = datetime.now()

        # 2.3查询时间段内的人数
        month_count = User.query.filter(User.last_login >= month_startTime_date,User.last_login<=month_endTime_date,User.is_admin==False).count()
    except Exception as e:
        current_app.logger.error(e)

    # 3,统计日活人数
    try:
        # 2.1本日０点
        day_startTime_str = "%d-%d-%d"%(cal.tm_year,cal.tm_mon,cal.tm_mday)

        # strptime() 参数１：表示时间字符串　　参数２：表示时间格式
        day_startTime_date = datetime.strptime(day_startTime_str,"%Y-%m-%d")

        # 2.2　此刻时间
        day_endTime_date = datetime.now()

        # 2.3查询时间段内的人数
        day_count = User.query.filter(User.last_login >= day_startTime_date,User.last_login<=day_endTime_date,User.is_admin==False).count()
    except Exception as e:
        current_app.logger.error(e)

    # 4,时间段内活跃人数
    active_date = [] #活跃的日期
    active_count = [] #获取人数
    for i in range(0, 31):
        # 当天开始时间A
        begin_date = day_startTime_date - timedelta(days=i)
        # 当天开始时间, 的后一天B
        end_date = day_startTime_date - timedelta(days=i - 1)

        # 添加当天开始时间字符串到, 活跃日期中
        active_date.append(begin_date.strftime("%m-%d"))

        # 查询时间A到B这一天的注册人数
        everyday_active_count = User.query.filter(User.is_admin == False, User.last_login >= begin_date,
                                                  User.last_login <= end_date).count()

        # 添加当天注册人数到,获取数量中
        active_count.append(everyday_active_count)

    #为了方便查看，反转字符串
    active_date.reverse()
    active_count.reverse()

    # 5,携带参数,渲染页面
    data = {
        "total_count":total_count,
        "month_count":month_count,
        "day_count":day_count,
        "active_date":active_date,
        "active_count":active_count
    }

    return render_template("admin/user_count.html",data = data)

#显示管理员首页页面
# 请求路径: /admin/index
# 请求方式: GET
# 请求参数: 无
# 返回值:渲染页面index.html,user字典数据
@admin_blue.route('/index')
@user_login_data
def index():
    admin = g.user.to_dict() if g.user else ""
    return render_template("admin/index.html",admin=admin)

# 显示登陆界面
# 请求路径: /admin/login
# 请求方式: GET,POST
# 请求参数:GET,无, POST,username,password
# 返回值: GET渲染login.html页面, POST,login.html页面,errmsg
@admin_blue.route('/login',methods=["GET","POST"])
def admin_login():
    """
    1,如果是GET方式，直接返回到首页
    2,如果是POST方式，获取参数
    3,校验参数，为空校验
    4,通过用户名，查询管理员对象，并判断用户是否存在
    5,判断密码是否正确
    6,记录管理员的session信息
    7,重定向到首页
    :return:
    """
    # 1,如果是GET方式，直接返回到首页
    if request.method == "GET":

        # 判断管理员是否登录
        if session.get("is_admin"):
            return redirect("admin/index")

        return  render_template("admin/login.html")

    # 2,如果是POST方式，获取参数
    username = request.form.get("username")
    password = request.form.get("password")

    # 3,校验参数，为空校验
    if not all([username,password]):
        return render_template("admin/login.html",errmsg = "参数不全")

    # 4,通过用户名，查询管理员对象，并判断用户是否存在
    try:
       admin = User.query.filter(User.mobile == username,User.is_admin == True ).first()
    except Exception as e:
        return  render_template("admin/login.html",errmsg = "获取管理员失败")

    if not admin:
        return render_template("admin/login.html",errmsg = "管理员不存在")

    # 5,判断密码是否正确
    if not password:
        return render_template("admin/login.html",errmsg = "密码不正确")

    # 6,记录管理员的session信息
    session["user_id"] = admin.id
    session["nick_name"] = admin.nick_name
    session["mobile"] = admin.mobile
    session["is_admin"] = admin.is_admin

    # 7,重定向到首页
    return redirect("admin/index")
